Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

cache: convert prune loop from recursive invocation to iterative #5636

Merged
merged 1 commit into from
Jan 16, 2025

Conversation

jsternberg
Copy link
Collaborator

The prune logic would prune multiple times because one prune could cause more things to be capable of pruning and change the logic. This was done through a recursive invocation.

Since go doesn't have support for function tail calls, this would result in a new stack entry for each loop. This unrolls the logic so the prune function is invoked iteratively rather than recursively.

prune and pruneOnce have also had their names swapped. In general, pruneOnce implies that it runs a single prune while prune sounds like the top level function. The current code had this reversed and pruneOnce would call prune and prune would call itself recursively.

I've also updated a section in the controller that invoked prune on each worker. In older versions of Go, the current version was correct because those versions of Go would reuse the location for each loop which would cause goroutines to all reference the same worker instead of different workers.

Recent versions of Go have changed the behavior so this is no longer needed.

@github-actions github-actions bot added area/hack building buildkit itself area/storage area/solver labels Jan 9, 2025
@jsternberg jsternberg added this to the v0.19.0 milestone Jan 9, 2025
@jsternberg jsternberg requested a review from tonistiigi January 9, 2025 19:30
cache/manager.go Outdated
return err
}
totalReleasedSize += releasedSize
totalReleasedCount += releasedCount
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Where are these used?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

These were used for the tracing part in the original PR. I'll remove them from this and readd them there.

})
}
for {
releasedSize, releasedCount, err := cm.pruneOnce(ctx, ch, popt)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

releasedCount in here seems more like a bool indicating if anything was pruned (unless I'm missing some usage of totalReleasedCount).

return nil
})
}(w)
eg.Go(func() error {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I guess this is go1.22+ only

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Our go.mod says go1.22 so I just changed this while I was looking at it since this isn't needed anymore.

The prune logic would prune multiple times because one prune could cause
more things to be capable of pruning and change the logic. This was done
through a recursive invocation.

Since go doesn't have support for function tail calls, this would result
in a new stack entry for each loop. This unrolls the logic so the prune
function is invoked iteratively rather than recursively.

`prune` and `pruneOnce` have also had their names swapped. In general,
`pruneOnce` implies that it runs a single prune while `prune` sounds
like the top level function. The current code had this reversed and
`pruneOnce` would call `prune` and `prune` would call itself
recursively.

I've also updated a section in the controller that invoked prune on each
worker. In older versions of Go, the current version was correct because
those versions of Go would reuse the location for each loop which would
cause goroutines to all reference the same worker instead of different
workers.

Recent versions of Go have changed the behavior so this is no longer
needed.

Signed-off-by: Jonathan A. Sternberg <[email protected]>
@tonistiigi tonistiigi merged commit e7185ac into moby:master Jan 16, 2025
97 checks passed
@jsternberg jsternberg deleted the unroll-prune-loop branch January 17, 2025 15:33
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants